# Experiment 6-7 String Recognizer using the Krypton CPLD

Dhruv Ilesh Shah — 150070016 March 3, 2017

#### Overview

Sequential circuits are an important part of digital logic, and hence, in this report, I present the complete implementation of a *string recognizer* that can identify the occurrence of the following words.

- bomb
- gun
- knife
- terror

The code was compiled on Quartus Prime, and simulated using ModelSim. GHDL was also used for simulation purposes, at a low level. This was then uploaded to the  $Krypton\ v1.1$  5M1270ZT144C5N CPLD-based board.

The codes and setup have been covered in section 1. We build the string recognizer piecewise, by implementing each module independently. The VHDL codes have been kept modular and as generic as possible, for reusability and code clarity. Section 2 presents the simulation observations and miscellaneous results. Section 3 presents the observations after running the scan-chain test on the board.

# 1 Setup

The english alphabet can be represented by 5 bits ( $\lceil log_2(26) \rceil$ ), and a bit each for *clock* and *reset* results in 7 input bits. Managing all these, along with keeping track of the states would actually be a tedious task and we would have to deal with a complex machine with close to **300** states! That is where the idea of interacting FSMs comes handy. In this report, I implement individual (and independent) FSMs for detecting each of the strings, and an OR of the four outputs gives the required output bit.

The inputs were encoding by converting alphabet position (1-26) to 5-bit binary, for simplicity. The various entities used for the detection include *bomber*, *gunman*, *knife\_hurler & terrorist*. The output can be represented with one bit, 1 standing for positive (string detected!).

#### 1.1 Bomber

Straight from the description, a description of the problem in terms of a finite-state machine is given below.



Figure 1: Automata Representation for Bomber

As a notation in the above figure, the text on each edge of the graph a/b represents input a to the machine, and output b of the machine. State encoding can be done in a smart way to reduce the combinational overhead for each state bit. For this purpose, I have used  $Gray\ Codes$  to encode the states.

| State  | $q_1$ | $q_0$ |
|--------|-------|-------|
| $\phi$ | 0     | 1     |
| b      | 1     | 1     |
| bo     | 1     | 0     |
| bom    | 0     | 0     |

Following the above state assignment and FSM design (Figure 1), the code for detecting bomb is given below.

```
library ieee;
   use ieee.std_logic_1164.all;
   library work;
   use work. EE224_Components.all;
4
5
   entity bomber is
6
            port(x: in std_ulogic_vector(4 downto 0);
7
                    reset, clk: in std_ulogic;
8
                    s: out std_ulogic);
9
   end entity;
10
11
   architecture bomb_detector of bomber is
12
            signal q, nq, qb: std_ulogic_vector(1 downto 0);
13
            signal b, b1, b2, b3, n0, n1, n2, n3, n4, bb, q01, q02, q03,
14
           phis, bs, bos, boms, m, m1, m2, m3, mb, q11, q12, q13, o, ob: std_ulogic;
15
   --State Description:
16
   --phi
                          0 1
17
                         1 1
   --Ъ
18
   −−bo
                          1 0
19
```

```
0 0
   --bom
20
21
   begin
22
            INVO: inverter port map (x(0), n0);
23
            N100: inverter port map (x(1), n1);
24
            INV2: inverter port map (x(2), n2);
25
            INV3: inverter port map (x(3), n3);
26
            INV4: inverter port map (x(4), n4);
27
            A5b: andi5 port map (n4, n3, n2, x(1), n0, b); -- Identifying B
29
            A5m: and i5 port map (n4, x(3), x(2), n1, x(0), m); -- Identifying M
30
            A5o: and i5 port map (n4, x(3), x(2), x(1), x(0), o); -- Identifying 0
31
32
            INV5: inverter port map (b, bb);
34
            INV6: inverter port map (m, mb);
35
            INV60: inverter port map (o, ob);
36
37
            -- Deciphering states
38
            N7: inverter port map (q(0), qb(0));
            N8: inverter port map (q(1), qb(1));
40
41
            A5: andi2 port map (qb(1), q(0), phis);
42
            A6: andi2 port map (q(1), q(0), bs);
43
            A7: andi2 port map (q(1), qb(0), bos);
            A8: andi2 port map (qb(1), qb(0), boms);
45
46
            -- Assigning state nq(0)
47
            A9: andi2 port map (boms, b, q01);
48
            01: ori2 port map (phis, q01, q02);
49
            A10: andi2 port map (bs, ob, q03);
50
            O2: ori2 port map (qO2, qO3, nq(0));
51
52
            -- Assigning state nq(1)
53
            A11: andi2 port map (phis, b, q11);
54
            A12: andi2 port map (bos, mb, q12);
55
            03: ori2 port map (q11, q12, q13);
56
            04: ori2 port map (q13, bs, nq(1));
58
            -- Assigning the Output
59
            A13: andi2 port map (boms, b, s);
60
61
            -- Adding dffi's
62
            d1: dffi port map (d \Rightarrow nq(1), clk \Rightarrow clk, q \Rightarrow q(1), r \Rightarrow reset);
63
            d0: dffi1 port map (d \Rightarrow nq(0), clk \Rightarrow clk, q \Rightarrow q(0), r \Rightarrow reset);
64
65
   end bomb_detector;
66
```

#### 1.2 Gunman



Figure 2: Automata Representation for Gunman

Going ahead with the above state representation, the encoding has been done as shown below (since only 2 non- $\phi$  states, one-hot encoding serves the purpose).

| State         | $q_1$ | $q_0$ |
|---------------|-------|-------|
| $\phi$        | 0     | 0     |
| $\mid g \mid$ | 1     | 0     |
| gu            | 0     | 1     |

Following the above state assignment and FSM design (Figure 2), the code for detecting gun is given below.

```
library ieee;
   use ieee.std_logic_1164.all;
3
   library work;
   use work.EE224_Components.all;
4
   entity gunman is
6
            port(x: in std_ulogic_vector(4 downto 0);
7
                    reset, clk: in std_ulogic;
8
                    s: out std_ulogic);
9
   end entity;
10
11
   architecture gun_detector of gunman is
^{12}
            signal q, nq, qb: std_ulogic_vector(1 downto 0);
13
            signal n0, n1, n2, n3, n4, g, u, n, gb, nb, ub, gs,
14
            gus, phis, g11, g12, g01, g02: std_ulogic;
15
   --State Description
16
                           0 0
   --phi
17
                         1 0
   --g
18
                          0 1
   --gu
19
   begin
20
```

```
INVO: inverter port map (x(0), n0);
21
            INV1: inverter port map (x(1), n1);
22
            INV2: inverter port map (x(2), n2);
23
            INV3: inverter port map (x(3), n3);
24
            INV4: inverter port map (x(4), n4);
25
26
            A51: and i5 port map (n4, n3, x(2), x(1), x(0), g); -- Identifying G
27
            A52: and i5 port map (x(4), n3, x(2), n1, x(0), u); -- Identifying U
            A53: and i5 port map (n4, x(3), x(2), x(1), n0, n); -- Identifying N
29
30
            INV5: inverter port
                                           map (q(0), qb(0));
31
            INV6: inverter port map (q(1), qb(1));
32
            INV7: inverter port map (g, gb);
33
            INV8: inverter port map (u, ub);
34
35
            INV9: inverter port map (n, nb);
36
37
            -- Declaring the States
38
            A1: andi2 port map (qb(0), qb(1), phis);
            A2: andi2 port map (qb(0), q(1), gus);
            A3: andi2 port map (qb(1), q(0), gs);
41
42
            -- Assigning State nq(0)
43
            A4: andi2 port map (phis, g, g11);
44
            A5: andi2 port map (gs, ub, g12);
            O1: ori2 port map (g11, g12, nq(0));
46
47
            -- Assigning State nq(1)
48
            A6: andi2 port map (gs, u, g01);
49
            A7: andi2 port map (gus, nb, g02);
50
            02: ori2 port map (g01, g02, nq(1));
51
52
            -- Assigning the Output
53
            A8: andi2 port map (gus, n, s);
54
55
             -- Adding the dffi's
56
            d0: dffi port map (d \Rightarrow nq(0), clk \Rightarrow clk, q \Rightarrow q(0), r \Rightarrow reset);
            d1: dffi port map (d \Rightarrow nq(1), clk \Rightarrow clk, q \Rightarrow q(1), r \Rightarrow reset);
59
   end gun_detector;
60
```

# 1.3 Knife Hurler



Figure 3: Automata Representation for Knife Hurler

Going with the above state representation, the encoding requires 3 bits (5 states), and hence I have again followed *Gray Codes* for the encoding. This is shown below.

| State  | $q_2$ | $q_1$ | $q_0$ |
|--------|-------|-------|-------|
| $\phi$ | 0     | 0     | 0     |
| k      | 0     | 0     | 1     |
| kn     | 0     | 1     | 1     |
| kni    | 0     | 1     | 0     |
| knif   | 1     | 1     | 0     |

Following the above state assignment and FSM design (Figure 3), the code for detecting knife is given below.

```
library ieee;
   use ieee.std_logic_1164.all;
   library work;
3
   use work.EE224_Components.all;
4
5
   entity knife_hurler is
           port(x: in std_ulogic_vector(4 downto 0);
8
                    reset, clk: in std_ulogic;
9
                    s: out std_ulogic);
10
   end entity;
11
12
   architecture knife_detector of knife_hurler is
13
            signal q, nq, qb: std_ulogic_vector(2 downto 0);
14
            signal k, n, i, f, e, kb, nb, ib, fb, eb, ks, phis,
15
```

```
kns, knis, knifs, n0, n1, n2, n3, n4,
16
            q01, q02, q03, q11, q12, q13, q14, q21, q22: std_ulogic;
17
    --State Description
18
    --phi
                          0 0 0
19
    --k
                         0 0 1
20
   --kn
                          0 1 1
21
                          0 1 0
   --kni
22
                           1 1 0
   --knif
23
24
   begin
25
            INVO: inverter port map (x(0), n0);
26
            INV1: inverter port map (x(1), n1);
27
            INV2: inverter port map (x(2), n2);
28
            INV3: inverter port map (x(3), n3);
29
            INV4: inverter port map (x(4), n4);
30
31
            A5k: and i5 port map (n4, x(3), n2, x(1), x(0), k); -- Identifying K
32
            A5n: and i5 port map (n4, x(3), x(2), x(1), n0, n); -- Identifying N
33
            A5i: andi5 port map (n4, x(3), n2, n1, x(0), i); -- Identifying I
34
            A5f: and i5 port map (n4, n3, x(2), n1, x(0), e); -- Identifying E
            A5e: and i5 port map (n4, n3, x(2), x(1), n0, f); -- Identifying F
37
            INV5: inverter port map (q(0), qb(0));
38
            INV6: inverter port map (q(1), qb(1));
39
            INV7: inverter port map (q(2), qb(2));
            INV8: inverter port map (k, kb);
41
            INV9: inverter port map (n, nb);
42
            INV10: inverter port map (i, ib);
43
            INV11: inverter port map (f, fb);
44
            INV12: inverter port map (e, eb);
45
46
            -- Declaring the states
47
            A31: andi3 port map (qb(2), qb(1), qb(0), phis);
48
            A32: and i3 port map (qb(2), qb(1), q(0), ks);
49
            A33: andi3 port map (qb(2), q(1), q(0), kns);
50
            A34: andi3 port map (qb(2), q(1), qb(0), knis);
51
            A35: andi3 port map (q(2), q(1), qb(0), knifs);
52
            -- Assigning State ng(2)
54
            A1: andi2 port map (knis, f, q21);
55
            A2: andi2 port map (knifs, eb, q22);
56
            O1: ori2 port map (q21, q22, nq(2));
57
58
            -- Assigning State nq(1)
59
            02: ori2 port map (kns, knis, q11);
60
            A3: andi2 port map (ks, n, q12);
61
            A4: andi2 port map (knifs, eb, q13);
62
            O3: ori2 port map (q12, q13, q14);
63
            04: ori2 port map (q11, q14, nq(1));
65
```

```
-- Assigning State nq(0)
66
              A5: andi2 port map (kns, ib, q01);
67
              A6: andi2 port map (phis, k, q02);
68
              05: ori2 port map (q02, ks, q03);
69
              O6: ori2 port map (q03, q01, nq(0));
70
71
              -- Assigning the Output
72
              A7: andi2 port map (knifs, e, s);
73
              -- Adding the dffi's
75
              d0: dffi port map (d \Rightarrow nq(0), clk \Rightarrow clk, q \Rightarrow q(0), r \Rightarrow reset);
76
              d1: dffi port map (d \Rightarrow nq(1), clk \Rightarrow clk, q \Rightarrow q(1), r \Rightarrow reset);
77
              d2: dffi port map (d \Rightarrow nq(2), clk \Rightarrow clk, q \Rightarrow q(2), r \Rightarrow reset);
78
79
    end knife_detector;
80
```

#### 1.4 Terrorist



Figure 4: Automata Representation for Terrorist

Going with the above state representation, the encoding requires 3 bits (6 states), and hence I have, yet again, followed *Gray-like Codes* for encoding the states.

| State  | $q_2$ | $q_1$ | $q_0$ |
|--------|-------|-------|-------|
| $\phi$ | 0     | 0     | 0     |
| t      | 0     | 0     | 1     |
| te     | 0     | 1     | 1     |
| ter    | 0     | 1     | 0     |
| terr   | 1     | 1     | 0     |
| terro  | 1     | 0     | 0     |

Following the above state assignment and FSM design (Figure 4), the code for detecting terror is given below.

```
library ieee;
   use ieee.std_logic_1164.all;
   library work;
   use work. EE224_Components.all;
4
   entity terrorist is
            port(x: in std_ulogic_vector(4 downto 0);
8
                    reset, clk: in std_ulogic;
9
                    s: out std_ulogic);
10
   end entity;
11
12
   architecture terror_detector of terrorist is
13
            signal q, nq, qb: std_ulogic_vector(2 downto 0);
14
            signal t, e, r, o, tb, eb, rb, ob, phis, ts, tes, ters,
15
            terrs, terros, n0, n1, n2, n3, n4, q01, q02, q03, q11,
16
            q12, q13, q14, q21, q22, q23: std_ulogic;
17
   --State Description
19
   --phi
                 000
20
   --t
                        0 0 1
21
   --te
                0 1 1
22
   --ter
                 0 1 0
                  1 1 0
   --terr
   --terro
                   1 0 0
25
26
   begin
27
            INVO: inverter port map (x(0), n0);
28
            INV1: inverter port map (x(1), n1);
            INV2: inverter port map (x(2), n2);
30
            INV3: inverter port map (x(3), n3);
31
            INV4: inverter port map (x(4), n4);
32
33
            A5e: andi5 port map (n4, n3, x(2), n1, x(0), e); -- Identifying E
34
            A5o: and i5 port map (n4, x(3), x(2), x(1), x(0), o); -- Identifying O
35
            A5t: and i5 port map (x(4), n3, x(2), n1, n0, t); -- Identifying T
36
            A5r: and i5 port map (x(4), n3, n2, x(1), n0, r);
37
38
            INV5: inverter port
                                        map (q(0), qb(0));
39
                                        map (q(1), qb(1));
            INV6: inverter port
40
```

```
INV7: inverter port map (q(2), qb(2));
41
             INV8: inverter port map (t, tb);
42
             INV9: inverter port map (e, eb);
43
             INV10: inverter port map (r, rb);
44
             INV11: inverter port map (o, ob);
45
46
             -- Declaring the states
47
             A31: and i3 port map (qb(2), qb(1), qb(0), phis);
             A32: and i3 port map (qb(2), qb(1), q(0), ts);
49
             A33: andi3 port map (qb(2), q(1), q(0), tes);
50
             A34: andi3 port map (qb(2), q(1), qb(0), ters);
51
             A35: andi3 port map (q(2), q(1), qb(0), terrs);
52
             A36: andi3 port map (q(2), qb(1), qb(0), terros);
53
54
             -- Assigning State nq(2)
55
             A1: andi2 port map (ters, r, q21);
56
             A2: andi2 port map (terros, rb, q22);
57
             O1: ori2 port map (q21, q22, q23);
58
             O11: ori2 port map (q23, terrs, nq(2));
59
             -- Assigning State ng(1)
62
             02: ori2 port map (tes, terrs, q11);
63
             A3: andi2 port map (ts, e, q12);
64
             A4: andi2 port map (terrs, rb, q13);
             03: ori2 port map (q12, q13, q14);
             O4: ori2 port map (q11, q14, nq(1));
67
68
             -- Assigning State ng(0)
69
             A5: andi2 port map (tes, rb, q01);
70
             A6: andi2 port map (phis, t, q02);
71
             05: ori2 port map (q02, ts, q03);
72
             O6: ori2 port map (q03, q01, nq(0));
73
74
             -- Assigning the Output
75
             A7: andi2 port map (terros, r, s);
76
77
             -- Adding the dffi's
             d0: dffi port map (d \Rightarrow nq(0), clk \Rightarrow clk, q \Rightarrow q(0), r \Rightarrow reset);
79
             d1: dffi port map (d \Rightarrow nq(1), clk \Rightarrow clk, q \Rightarrow q(1), r \Rightarrow reset);
80
             d2: dffi port map (d \Rightarrow nq(2), clk \Rightarrow clk, q \Rightarrow q(2), r \Rightarrow reset);
81
82
   end terror_detector;
```

#### Bringing it all Together

Now that we have 4 independent FSMs that seem to be doing their job right, we must put them all together, in order to obtain the machine as required in the problem description. Without much effort, this can be done by simply taking an OR operation of the 4 individual outputs. This has been done in the DUT.

```
library ieee;
   use ieee.std_logic_1164.all;
3
   entity DUT is
4
    port(input_vector: in std_ulogic_vector(6 downto 0);
5
            output_vector: out std_ulogic_vector(0 downto 0));
   end entity;
   architecture WRAP of DUT is
9
   component bomber is
10
      port(x: in std_ulogic_vector(4 downto 0);
11
            reset, clk: in std_ulogic;
12
            s: out std_ulogic);
13
   end component;
14
15
   component gunman is
16
            port(x: in std_ulogic_vector(4 downto 0);
17
                     reset, clk: in std_ulogic;
18
                     s: out std_ulogic);
19
   end component;
20
21
   component knife_hurler is
22
            port(x: in std_ulogic_vector(4 downto 0);
23
                     reset, clk: in std_ulogic;
24
                     s: out std_ulogic);
25
   end component;
26
27
   component terrorist is
28
            port(x: in std_ulogic_vector(4 downto 0);
29
                     reset, clk: in std_ulogic;
30
                     s: out std_ulogic);
31
   end component;
32
33
      signal res, bs, gs, ks, ts, clk: std_ulogic;
34
      signal x: std_ulogic_vector(4 downto 0);
35
   begin
36
            res <= input_vector(6);</pre>
37
            clk <= input_vector(5);</pre>
39
            output_vector(0) <= bs or gs or ks or ts;
40
            x <= input_vector(4 downto 0);</pre>
41
42
            B: bomber port map (reset \Rightarrow res, x \Rightarrow x, s \Rightarrow bs, clk \Rightarrow clk);
43
            G: gunman port map (reset => res, x => x, s => gs, clk => clk);
44
            K: knife_hurler port map (reset => res, x => x, s => ks, clk => clk);
45
            T: terrorist port map (reset => res, x => x, s => ts, clk => clk);
46
   end WRAP;
47
```

# 2 Observations

After implementing the design in code, the next major part is to simulate and test the code for a set of inputs. RTL and Gate-Level simulation was performed on the machine, as a whole. Snapshots of the same are given in Figures 5-8. The validity of the code can be ascertained by the fact that all test cases passed successfully.



Figure 5: RTL Simulation of the String Detector



Figure 6: Gate-level Simulation of the String Detector



Figure 7: Gate-level Simulation Zoomed-in to a smaller time frame

# 3 Scan-Chain Tests

We have tested the logic using the RTL simulations, emulated the CPLD performance using the gate-level simulation and uploaded the code on the Krypton board. Next, we need to check that the code is actually running as it is expected to, on the board. We could do so manually but that is not feasible due to the following reasons.

- Our current circuit requires 7 control switches, including a clock. In any given setup, it may not be possible to allocate as many I/O pins. As the complexity increases, it will indeed not be possible to allocate so many pins.
- Even if the above is possible, the total number of test cases is *exponential* in the size of the input and it is impractical to perform each of this manually.<sup>1</sup>

Hence, we test the uploaded code on the hardware using the scan-chain setup, as suggested in the manual. This setup was run on a set of two collections of text which has occurrences of the concerned string.

<sup>&</sup>lt;sup>1</sup>It is not wise to skip any case because, say, we do miss out a failed case it can cascade into unimaginable consequences, which can become difficult to debug.

#### Results

```
66087 : SDR 18 TDI(101FE) 8 TDO(03) MASK(FF) -----#
Successfully entered the input..
Sampling out data..
 ---Success for FF
Output Comparison :
                      Success
#----- Command - 66088 : RUNTEST 1 MSEC -----#
                   66089 : SDR 18 TDI(201FE) 8 TDO(00) MASK(FF) -----#
 ----- Command -
Successfully entered the input..
Sampling out data..
----Success for 03
Output Comparison : Success
     -- Command - 66090 : RUNTEST 1 MSEC -----#
  ---- Command - 66091 : SDR 18 TDI(301FE) 8 TDO(00) MASK(FF) -----#
Successfully entered the input..
Sampling out data..
 ---Success for 00
Output Comparison : Success
  ---- Command - 66092 : RUNTEST 1 MSEC -----#
Sampling out data..
 ---Success for 00
Output Comparison : Success
K. All Test Cases Passed.
ransaction Complete.
```

Figure 8: Results of the scan-chain test for random sample test cases.

# Conclusion

Starting from the very scratch, in this report, I have presented the logic and code for a sequential implementation of a string recognizer. The logic was tested using RTL simulation, followed by the gate-level simulation for delay analysis and emulating the CPLD. This was followed by an actual rigorous test on the CPLD board after burning the code on it, using the TIVA-C microcontroller.

All the cases passed successfully at all stages and hence the complete string recognizer can be used in hardware, as required.